#pragma once

#define NULL_MSG_LEN -1
/*
 * kiwi.
 *
 * postgreSQL protocol interaction library.
 */

KIWI_API static inline machine_msg_t *
kiwi_be_write_error_as(machine_msg_t *msg, char *severity, char *code,
		       char *detail, int detail_len, char *hint, int hint_len,
		       char *message, int len)
{
	size_t size = 1 /* S */ + 6 + 1 /* C */ + 6 + 1 /* M */ + len + 1 +
		      1 /* zero */;
	if (detail && detail_len > 0) {
		size += 1 + /* D */ +detail_len + 1;
	}
	if (hint && hint_len > 0) {
		size += 1 + /* H */ +hint_len + 1;
	}
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, sizeof(kiwi_header_t) + size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_ERROR_RESPONSE);
	kiwi_write32(&pos, sizeof(uint32_t) + size);
	kiwi_write8(&pos, 'S');
	kiwi_write(&pos, severity, 6);
	kiwi_write8(&pos, 'C');
	kiwi_write(&pos, code, 6);
	if (detail && detail_len > 0) {
		kiwi_write8(&pos, 'D');
		kiwi_write(&pos, detail, detail_len);
		kiwi_write8(&pos, 0);
	}
	if (hint && hint_len > 0) {
		kiwi_write8(&pos, 'H');
		kiwi_write(&pos, hint, hint_len);
		kiwi_write8(&pos, 0);
	}
	kiwi_write8(&pos, 'M');
	kiwi_write(&pos, message, len);
	kiwi_write8(&pos, 0);
	kiwi_write8(&pos, 0);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_error(machine_msg_t *msg, char *code, char *message, int len)
{
	return kiwi_be_write_error_as(msg, "ERROR", code, NULL, 0, NULL, 0,
				      message, len);
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_error_fatal(machine_msg_t *msg, char *code, char *message,
			  int len)
{
	return kiwi_be_write_error_as(msg, "FATAL", code, NULL, 0, NULL, 0,
				      message, len);
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_error_panic(machine_msg_t *msg, char *code, char *message,
			  int len)
{
	return kiwi_be_write_error_as(msg, "PANIC", code, NULL, 0, NULL, 0,
				      message, len);
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_notice_as(machine_msg_t *msg, char *severity, int severity_len,
			char *code, char *detail, int detail_len, char *hint,
			int hint_len, char *message, int len)
{
	size_t size = 1 /* S */ + severity_len + 1 /* C */ + 6 + 1 /* M */ +
		      len + 1 + 1 /* zero */;

	if (detail && detail_len > 0) {
		size += 1 + /* D */ +detail_len + 1;
	}
	if (hint && hint_len > 0) {
		size += 1 + /* H */ +hint_len + 1;
	}
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, sizeof(kiwi_header_t) + size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_NOTICE_RESPONSE);
	kiwi_write32(&pos, sizeof(uint32_t) + size);
	kiwi_write8(&pos, 'S');
	kiwi_write(&pos, severity, severity_len);
	kiwi_write8(&pos, 'C');
	kiwi_write(&pos, code, 6);
	if (detail && detail_len > 0) {
		kiwi_write8(&pos, 'D');
		kiwi_write(&pos, detail, detail_len);
		kiwi_write8(&pos, 0);
	}
	if (hint && hint_len > 0) {
		kiwi_write8(&pos, 'H');
		kiwi_write(&pos, hint, hint_len);
		kiwi_write8(&pos, 0);
	}
	kiwi_write8(&pos, 'M');
	kiwi_write(&pos, message, len);
	kiwi_write8(&pos, 0);
	kiwi_write8(&pos, 0);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_notice_console_usage(machine_msg_t *msg, char *detail)
{
	return kiwi_be_write_notice_as(msg, "NOTICE", 7,
				       KIWI_SUCCESSFUL_COMPLETION, NULL, 0,
				       NULL, 0, detail, strlen(detail));
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_notice_info(machine_msg_t *msg, char *message, int len)
{
	return kiwi_be_write_notice_as(msg, "INFO", 4,
				       KIWI_SUCCESSFUL_COMPLETION, NULL, 0,
				       NULL, 0, message, len);
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_ok(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint32_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint32_t));
	kiwi_write32(&pos, 0);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_clear_text(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint32_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint32_t));
	kiwi_write32(&pos, 3);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_md5(machine_msg_t *msg, char salt[4])
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint32_t) + 4;
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint32_t) + 4);
	kiwi_write32(&pos, 5);
	kiwi_write(&pos, salt, 4);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_sasl(machine_msg_t *msg, char **mechanism,
				  size_t mechanism_cnt)
{
	size_t total_mechanism_len = 0;
	size_t mechanism_len;
	for (size_t i = 0; i < mechanism_cnt; ++i) {
		total_mechanism_len += strlen(mechanism[i]);
	}
	size_t size =
		sizeof(kiwi_header_t) + sizeof(uint32_t) + total_mechanism_len +
		mechanism_cnt *
			sizeof(uint8_t) /* zero-bytes at the end of each mechanism*/
		+ sizeof(uint8_t);

	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;

	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, size - sizeof(uint8_t));
	kiwi_write32(&pos, KIWI_BE_SASL_INIT);
	for (size_t i = 0; i < mechanism_cnt; ++i) {
		mechanism_len = strlen(mechanism[i]);
		kiwi_write(&pos, mechanism[i], mechanism_len);
		kiwi_write8(&pos, 0); /* write mechanism as a string */
	}
	kiwi_write8(&pos, 0); /* mimic the server response of PostgreSQL, 
	zero byte after last auth mechanism indicates that list ended */

	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_sasl_continue(machine_msg_t *msg, char *data,
					   int data_len)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint32_t) + data_len;

	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;

	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, size - sizeof(uint8_t));
	kiwi_write32(&pos, KIWI_BE_SASL_CONTINUE);
	kiwi_write(&pos, data, data_len);

	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_authentication_sasl_final(machine_msg_t *msg, char *data,
					int data_len)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint32_t) + data_len;

	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;

	kiwi_write8(&pos, KIWI_BE_AUTHENTICATION);
	kiwi_write32(&pos, size - sizeof(uint8_t));
	kiwi_write32(&pos, KIWI_BE_SASL_FINAL);
	kiwi_write(&pos, data, data_len);

	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_backend_key_data(machine_msg_t *msg, uint32_t pid, uint32_t key)
{
	size_t size =
		sizeof(kiwi_header_t) + sizeof(uint32_t) + sizeof(uint32_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_BACKEND_KEY_DATA);
	kiwi_write32(&pos,
		     sizeof(uint32_t) + sizeof(uint32_t) + sizeof(uint32_t));
	kiwi_write32(&pos, pid);
	kiwi_write32(&pos, key);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_parameter_status(machine_msg_t *msg, char *key, int key_len,
			       char *value, int value_len)
{
	size_t size = sizeof(kiwi_header_t) + key_len + value_len;
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_PARAMETER_STATUS);
	kiwi_write32(&pos, sizeof(uint32_t) + key_len + value_len);
	kiwi_write(&pos, key, key_len);
	kiwi_write(&pos, value, value_len);
	return msg;
}

KIWI_API static inline machine_msg_t *kiwi_be_write_ready(machine_msg_t *msg,
							  uint8_t status)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint8_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_READY_FOR_QUERY);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint8_t));
	kiwi_write8(&pos, status);
	return msg;
}

KIWI_API static inline int kiwi_be_write_complete(machine_msg_t *msg,
						  char *message, int len)
{
	size_t size = sizeof(kiwi_header_t) + len;
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	machine_msg_t *local_msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(local_msg == NULL)) {
		return -1;
	}
	char *pos;
	pos = (char *)machine_msg_data(local_msg) + offset;
	kiwi_write8(&pos, KIWI_BE_COMMAND_COMPLETE);
	kiwi_write32(&pos, sizeof(uint32_t) + len);
	kiwi_write(&pos, message, len);
	if (msg == NULL) {
		machine_msg_free(local_msg);
	}
	return 0;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_command_complete(machine_msg_t *msg, char *message, int len)
{
	size_t size = sizeof(kiwi_header_t) + len;
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_COMMAND_COMPLETE);
	kiwi_write32(&pos, sizeof(uint32_t) + len);
	kiwi_write(&pos, message, len);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_empty_query(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_EMPTY_QUERY_RESPONSE);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_parse_complete(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_PARSE_COMPLETE);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_bind_complete(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_BIND_COMPLETE);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_close_complete(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_CLOSE_COMPLETE);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_portal_suspended(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_PORTAL_SUSPENDED);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *kiwi_be_write_no_data(machine_msg_t *msg)
{
	size_t size = sizeof(kiwi_header_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_NO_DATA);
	kiwi_write32(&pos, sizeof(uint32_t));
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_row_description(machine_msg_t *msg, int *begin_offset)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint16_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	*begin_offset = offset;
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_ROW_DESCRIPTION);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint16_t));
	kiwi_write16(&pos, 0);
	return msg;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_compression_ack(machine_msg_t *msg, char compression_algorithm)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(char);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;
	kiwi_write8(&pos, KIWI_BE_COMPRESSION);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(char));
	kiwi_write8(&pos, compression_algorithm);
	return msg;
}

KIWI_API static inline int kiwi_be_write_row_description_add(
	machine_msg_t *msg, int begin_offset, char *name, int name_len,
	int32_t table_id, int16_t attrnum, int32_t type_id, int16_t type_size,
	int32_t type_modifier, int32_t format_code)
{
	assert(msg);
	size_t size = name_len + 1 + sizeof(uint32_t) /* table_id */ +
		      sizeof(uint16_t) /* attrnum */ +
		      sizeof(uint32_t) /* type_id */ +
		      sizeof(uint16_t) /* type_size */ +
		      sizeof(uint32_t) /* type_modifier */ +
		      sizeof(uint16_t) /* format_code */;

	size_t size_written = machine_msg_size(msg);
	int rc;
	rc = machine_msg_write(msg, NULL, size);
	if (kiwi_unlikely(rc == -1)) {
		return -1;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + size_written;
	kiwi_write(&pos, name, name_len);
	kiwi_write8(&pos, 0);
	kiwi_write32(&pos, table_id);
	kiwi_write16(&pos, attrnum);
	kiwi_write32(&pos, type_id);
	kiwi_write16(&pos, type_size);
	kiwi_write32(&pos, type_modifier);
	kiwi_write16(&pos, format_code);

	kiwi_header_t *header;
	header =
		(kiwi_header_t *)((char *)machine_msg_data(msg) + begin_offset);
	uint32_t pos_size = sizeof(uint32_t) + sizeof(uint16_t);
	pos = (char *)&header->len;
	uint32_t total_size;
	uint16_t count;
	kiwi_read32(&total_size, &pos, &pos_size);
	kiwi_read16(&count, &pos, &pos_size);
	total_size += size;
	count++;
	kiwi_write32to((char *)&header->len, total_size);
	kiwi_write16to((char *)&header->len + sizeof(uint32_t), count);
	return 0;
}

KIWI_API static inline machine_msg_t *
kiwi_be_write_row_descriptionf(machine_msg_t *msg, char *fmt, ...)
{
	int is_msg_allocated = msg == NULL;
	int begin_offset;
	msg = kiwi_be_write_row_description(msg, &begin_offset);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}

	va_list args;
	va_start(args, fmt);
	while (*fmt) {
		char *name = va_arg(args, char *);
		size_t name_len = strlen(name);
		int rc = 0;
		switch (*fmt) {
		case 's':
			rc = kiwi_be_write_row_description_add(
				msg, begin_offset, name, name_len, 0, 0,
				25 /* TEXTOID */, -1, 0, 0);
			break;
		case 'd':
			rc = kiwi_be_write_row_description_add(
				msg, begin_offset, name, name_len, 0, 0,
				23 /* INT4OID */, 4, 0, 0);
			break;
		case 'l':
			rc = kiwi_be_write_row_description_add(
				msg, begin_offset, name, name_len, 0, 0,
				20 /* INT8OID */, 8, 0, 0);
			break;
		case 'b':
			rc = kiwi_be_write_row_description_add(
				msg, begin_offset, name, name_len, 0, 0,
				16 /* BOOLOID */, 1, 0, 0);
			break;
		case 'f':
			rc = kiwi_be_write_row_description_add(
				msg, begin_offset, name, name_len, 0, 0,
				700 /* FLOAT4OID */, 4, 0, 0);
			break;
		}
		if (rc == -1) {
			if (is_msg_allocated) {
				machine_msg_free(msg);
			}
			va_end(args);
			return NULL;
		}
		fmt++;
	}
	va_end(args);
	return msg;
}

KIWI_API static inline machine_msg_t *kiwi_be_write_data_row(machine_msg_t *msg,
							     int *begin_offset)
{
	size_t size = sizeof(kiwi_header_t) + sizeof(uint16_t);
	int offset = 0;
	if (msg) {
		offset = machine_msg_size(msg);
	}

	*begin_offset = offset;
	msg = machine_msg_create_or_advance(msg, size);
	if (kiwi_unlikely(msg == NULL)) {
		return NULL;
	}

	char *pos;
	pos = (char *)machine_msg_data(msg) + offset;

	kiwi_write8(&pos, KIWI_BE_DATA_ROW);
	kiwi_write32(&pos, sizeof(uint32_t) + sizeof(uint16_t));
	kiwi_write16(&pos, 0);

	return msg;
}

KIWI_API static inline int kiwi_be_write_data_row_add(machine_msg_t *msg,
						      int begin_offset,
						      char *data, int32_t len)
{
	assert(msg);
	int is_null = len == -1;
	size_t size = sizeof(uint32_t) + (is_null ? 0 : len);

	size_t size_written = machine_msg_size(msg);
	int rc;
	rc = machine_msg_write(msg, NULL, size);
	if (kiwi_unlikely(rc == -1)) {
		return -1;
	}
	char *pos;
	pos = (char *)machine_msg_data(msg) + size_written;
	kiwi_write32(&pos, len);
	if (!is_null) {
		kiwi_write(&pos, data, len);
	}

	kiwi_header_t *header;
	header =
		(kiwi_header_t *)((char *)machine_msg_data(msg) + begin_offset);
	uint32_t pos_size = sizeof(uint32_t) + sizeof(uint16_t);
	pos = (char *)&header->len;
	uint32_t total_size;
	uint16_t count;
	kiwi_read32(&total_size, &pos, &pos_size);
	kiwi_read16(&count, &pos, &pos_size);
	total_size += size;
	count++;
	kiwi_write32to((char *)&header->len, total_size);
	kiwi_write16to((char *)&header->len + sizeof(uint32_t), count);
	return 0;
}
